/**
 * Book控制器模块
 * 
 * 说明：
 * 中间件处理路由请求：
 * 1. req.params --> 路由参数 /:id -> req.params.id
 * 2. req.query  --> url查询串 ?k1=v1&k2=v2 -> req.query.k1 & req.query.k2
 * 3. req.body   --> 数据对象 {k1:v1, k2:v2} -> req.body.k1, req.body.k2
 * 
 * 中间件进行异步封装（async/await），不会直接执行，需要进行函数立即执行的方式。
 */

const express = require("express");

const util = require("../common/util");

/**
 * @typedef {BookDB}
 */
const BookDB = require("../model/sqlite/book");

/**
 * 获取书籍信息
 * @param {express.Request} req 请求
 * @param {express.Response} res 响应
 */
function find(req, res) {
    (async function () { 
        let db = BookDB.getInstance();
        await db.connect();
        let result = await db.find(req.params.id);
        // await db.close();
        util.logFormat(`获取【${req.params.id}】书籍信息%O`, result);
        res.json(util.FormatJSONData(200, `获取书籍信息【${req.params.id}】`, result));
    })().catch(err => {
        util.err(err);
        res.json(util.FormatJSONData(500,"服务器发生意外情况，无法完成请求！"));
    });
}

/**
 * 获取书籍信息列表
 * @param {express.Request} req 请求
 * @param {express.Response} res 响应
 */
function findAll(req, res) {
    (async function () { 
        // 1.处理输入的数据
        // FIXME:输入数据的正确性校验，一致性校验，完整性校验
        let limit = req.query.limit ? req.query.limit : -1;
        let offset = req.query.offset ? req.query.offset : -1;
        // 2.访问数据
        let db = BookDB.getInstance();
        await db.connect();
        let result = await db.findAll(limit,offset);
        // await db.close();

        // 3.输出数据
        util.logFormat(`获取【limit=${limit}&offset=${offset}】书籍信息%O`, result);
        res.json(util.FormatJSONData(200, `获取书籍信息列表`, result));
    })().catch(err => {
        util.err(err);
        res.json(util.FormatJSONData(500,"服务器发生意外情况，无法完成请求！"));
    });
}

/**
 * 新增书籍信息
 * @param {express.Request} req 请求
 * @param {express.Response} res 响应
 */
function add(req, res) {
    (async function () { 
        //FIXME:req.body的数据安全性校验
        let book = {
            title:req.body.title,
            pic:req.body.pic,
            localPic:req.body.localPic,//FIXME:处理远程图片为本地图片
            author:req.body.author,
            publisher:req.body.publisher,
            producer:req.body.producer,
            subtitle:req.body.subtitle,
            originalTitle:req.body.originalTitle,
            translator:req.body.translator,
            pubdate:req.body.pubdate,
            pages:req.body.pages,
            price:req.body.price,
            binding:req.body.binding,
            series:req.body.series,
            isbn:req.body.isbn,
            intro:req.body.intro,
            doubanId:req.body.doubanId,
            createdTime:Date.now(),
            updatedTime:Date.now()
        };
        let db = BookDB.getInstance();
        await db.connect();
        // FIXME:插入数据前，考虑校验是否已存在ISBN
        let result = await db.add(book);
        // await db.close();
        util.log(`新增书籍信息：lastID->${result}`);
        res.json(util.FormatJSONData(201, `新增书籍信息`, { lastID: result }));
    })().catch(err => {
        util.err(err);
        res.json(util.FormatJSONData(500,"服务器发生意外情况，无法完成请求！"));
    });
}

/**
 * 更新书籍信息
 * @param {express.Request} req 请求
 * @param {express.Response} res 响应
 */
function update(req, res) {
    (async function () { 
        //FIXME:req.body的数据安全性校验
        let book = {
            id:req.params.id,
            title:req.body.title,
            pic:req.body.pic,
            localPic:req.body.localPic,//FIXME:处理远程图片为本地图片
            author:req.body.author,
            publisher:req.body.publisher,
            producer:req.body.producer,
            subtitle:req.body.subtitle,
            originalTitle:req.body.originalTitle,
            translator:req.body.translator,
            pubdate:req.body.pubdate,
            pages:req.body.pages,
            price:req.body.price,
            binding:req.body.binding,
            series:req.body.series,
            isbn:req.body.isbn,
            intro:req.body.intro,
            doubanId:req.body.doubanId,
            updatedTime:Date.now()
        };
        let db = BookDB.getInstance();
        await db.connect();
        let result = await db.update(book);
        // await db.close();
        util.log(`更新书籍信息：changes->${result}`);
        res.json(util.FormatJSONData(200, `更新书籍信息`, { changes: result }));
    })().catch(err => {
        util.err(err);
        res.json(util.FormatJSONData(500,"服务器发生意外情况，无法完成请求！"));
    });
}

/**
 * 删除书籍信息
 * @param {express.Request} req 请求
 * @param {express.Response} res 响应
 */
function remove(req, res) {
    (async function () { 
        // FIXME:数据合法性校验
        let bookId = req.params.id;
        let db = BookDB.getInstance();
        await db.connect();
        let result = await db.remove(bookId);
        // await db.close();
        util.log(`删除书籍信息：changes->${result}`);
        res.json(util.FormatJSONData(204, `删除书籍信息`, { changes: result }));
    })().catch(err => {
        util.err(err);
        res.json(util.FormatJSONData(500,"服务器发生意外情况，无法完成请求！"));
    });
}

/**
 * 获取书籍总数
 * @param {express.Request} req 请求
 * @param {express.Response} res 响应
 */
function getCount(req, res) {
    (async function () { 
        // FIXME:数据合法性校验
        let db = BookDB.getInstance();
        await db.connect();
        let result = await db.getCount();
        // await db.close();
        util.log("书籍总数：%O", result);
        res.json(util.FormatJSONData(200, `获取书籍总数`, result ));
    })().catch(err => {
        util.err(err);
        res.json(util.FormatJSONData(500,"服务器发生意外情况，无法完成请求！"));
    });
}

/**
 * 获取指定关键字的书籍信息列表
 * @param {express.Request} req 请求
 * @param {express.Response} res 响应
 */
function search(req, res) {
    (async function () { 
        // 1.处理输入的数据
        // FIXME:输入数据的正确性校验，一致性校验，完整性校验
        let q = req.query.q;
        let limit = req.query.limit ? req.query.limit : -1;
        let offset = req.query.offset ? req.query.offset : -1;
        // 2.访问数据
        let db = BookDB.getInstance();
        await db.connect();
        let result = await db.search(q, limit, offset);
        // await db.close();

        // 3.输出数据
        util.logFormat(`获取关键字为【${q}】【limit=${limit}&offset=${offset}】书籍信息%O`, result);
        res.json(util.FormatJSONData(200, `获取指定关键字的书籍信息列表`, result));
    })().catch(err => {
        util.err(err);
        res.json(util.FormatJSONData(500,"服务器发生意外情况，无法完成请求！"));
    });
}

/**
 * 获取指定ISBN的书籍信息
 * @param {express.Request} req 请求
 * @param {express.Response} res 响应
 */
function findByIsbn(req, res) {
    // FIXME:ISBN号的合法性校验
    (async function () { 
        let db = BookDB.getInstance();
        await db.connect();
        let result = await db.findByIsbn(req.params.isbn);
        // await db.close();
        util.logFormat(`获取【${req.params.id}】书籍信息%O`, result);
        res.json(util.FormatJSONData(200, `获取指定ISBN的书籍信息`, result));
    })().catch(err => {
        util.err(err);
        res.json(util.FormatJSONData(500,"服务器发生意外情况，无法完成请求！"));
    });
}

module.exports = {
    find,
    findAll,
    add,
    update,
    remove,
    getCount,
    search,
    findByIsbn
};